Ruby 进阶

在中国,任何区分, 入门级 与 老手级的程序员?

  1. 什么是单元测试 unit test. (针对 方法级别的测试) 往往使用框架.
  2. 什么是持续集成 每隔一段时间, 运行所有的单元测试. continuous integration CI. jenkins
  3. 什么是重构. refactoring 在不改变代码的行为的前提下, 改变代码的结构.

单元测试

单元测试很重要.在任何语言中都是.

不要用DEBUG,断点这些东东. 这些都是人肉的测试.

一般单元测试写好了,以后的情况完全就是自动化运行测试

单元测试也是一切自动化测试, 持续集成,重构的基础.

TestUnit

这个是最经典的测试框架.

# apple_test.rb
class Apple
  def color
    'red'
  end
end

require 'test/unit'
class AppleTest< Test::Unit::TestCase
  def test_color
    assert Apple.new.color == 'red'
  end
end

我们会得到结果:

$ ruby apple_test.rb
Run options:

# Running tests:
.

Finished tests in 0.000465s, 2151.9213 tests/s, 2151.9213 assertions/s.

1 tests, 1 assertions, 0 failures, 0 errors, 0 skips

TDD: 测试驱动开发

优点:

  1. 自然而然就写了单元测试了. 进行持续集成是水到渠成的.
  2. 能够催生出优秀的实现代码(软件的设计层面).

(为什么? 如何催生出来的? 我们是先考虑测试(我们此时是使用者), 使用者最知道如何使用, 才是使用者最喜欢的. 那么, 满足"使用者写出的测试"的实现代码, 就是最优设计.

国内, 95%的人, 不写单元测试

我们入乡随俗, 不要永远不写, 我们要写30%的测试.

Rails中, 我们只写controller, model的测试, 就够了.

其他的测试,不写了.

Rspec

(rpsec)[http://rspec.info/]是一个BDD框架. 它鼓励我们先写行为,再写测试. 例如:

# 下面代码仅是示意. 运行的话需要在rails环境内, 后面rails章节会有描述
describe Apple do
  it 'color shoud be red' do
    Apple.new.color.should == 'red'
  end
end

Rescue , exception

跟java一样,Ruby也会抛出异常,捕获异常.

# test_error.rb
def error
  raise 'I got error!'
end

error

运行它,会看到出错:

$ ruby test.rb
test.rb:2:in `error': I got error! (RuntimeError)
  from test.rb:5:in `<main>'

可以使用 begin..rescue ..end 来处理异常

  # test_rescue.rb
  def error
    begin
      raise 'I got error!'
    rescue Exception => e
      puts e
      puts e.backtrace
    end
  end

  error

运行,就可以看到异常的名字,以及出错的轨迹都打印出来了:

$ ruby test_rescue.rb
I got error!
test_rescue.rb:3:in `error'
test_rescue.rb:10:in `<main>'

block, Proc, lambda

block, proc, lambda都源自于函数式语言的概念,是函数式(functional progarmming)编程的体现。在函数式编程语言的世界中,这三种结构叫做闭包(closure)。 proc, lambda是可以单独调用的闭包,而block不可以。

换句话说,proc和lambda可以转化成为block,当做参数传递给方法,但是block只能接在方法调用后面,不可以单独调用。

函数式编程的核心是:函数即数据!

在ruby当中,block是接在方法调用(method call)后面的do...end或者使用花括号({})包起来的代码块(code block)会,可以认为成方法的参数(parameter)。如果使用的是花括号的方式,一定不要产生,认为它是C语言中的过程语句的幻觉。这是错误的认知,是新手经常会犯错的地方。

以下两种写法是等价的:

[1, 2, 3].each do |x|
  puts x
end
[1, 2, 3].each {|x| puts x}

proc对象:

proc = Proc.new {|x| puts x*2}
proc.call(2) # 4

lambda对象:

lam = lambda {|x| puts x*2}
lam.call(2) # 4

在使用block的地方使用proc和lambda:

[1, 2, 3].each {|x| puts x*2 } # 2, 4, 6
[1, 2, 3].each(&proc) # 2, 4, 6
[1, 2, 3].each(&lam)  # 2, 4, 6

lambda和proc的两个重要区别是:

  • Proc对象对参数数量检查不严格, lambda对象则要求参数的数量精确匹配
  • Proc对象和lambda对象对return关键字的处理不一样

在block 中,不要用 return 关键字. 如果要返回一个值,直接把它放在最后就可以了.例如:

result = [1,2,3].map { |e|
  puts "e is: #{e}"
  e * 3
}

# 结果是:
e is: 1
e is: 2
e is: 3
=> [3, 6, 9]

Module

module有两个典型的作用:

  • 提供namespace(开源代码中大量使用,比如Rails项目中使用的各种gem)
  • 取消多重继承代码的结构混乱(在ruby中,class可以认为是一个加强的module)

简单的使用方式为:

一个典型的module为:

module Fruit
  def self.test_method
    "this is a test method in module"
  end

  def taste
    'good'
  end
end

include用来包含一个module到class里面,module里面的普通方法会变成class里面的instance method:

class Apple
  include Fruit
end
puts Apple.new.taste
# => good

extend把Module中的普通方法方法变成 "class method"

class Orange
  extend Fruit
end

puts Orange.taste
# => good

Class的继承

class Parent
  def family_name
    'Green'
  end
end

class Child < Parent
end

puts Child.new.family_name

# => Green

self

在ruby中,self表示当前作用域的 method caller. 可以认为是调用方法的对象.例如, :

temp_string = '1,2,3'

# 下面的`temp_string`就是一个caller
temp_string.split ','

为了方便记忆,只要见到 a.b() 这个形式,就可以说 a 就是 "method caller", 在方法b中, self就指代"a", 例如,下面的例子,我们可以把self打印出来:

class Apple
  def print_self
    puts self.inspect
  end
end

Apple.new.print_self
# => #<Apple:0x00000001f0dad8>

上面的 #<Apple:0x00000001f0dad8>就是 Apple.new 出来的实例. 

常用技巧

# 假设User有个属性,叫name
class User
  attr_accessor :name
end

jim = User.new
jim.name = 'Jim'

li_lei = User.name
li_lei.name = 'Li Lei'

那么, 下面的代码

[jim, li_lei].map { |user|
  user.name
}

等同于这种格式.(该格式常见于所有的开源项目中)

[jim, li_lei].map(&:name)

results matching ""

    No results matching ""